<!DOCTYPE html>


  <html class="dark page-post">


<head><meta name="generator" content="Hexo 3.9.0">
  <meta charset="utf-8">
  
  <title>webpack学习总结(二) | Poetry&#39;s Blog</title>

  <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">

  
    <meta name="keywords" content="webpack,">
  

  <meta name="description" content="声明 本教程整理于互联网   webpack是当下最热门的前端资源模块化管理和打包工具。它可以将许多松散的模块按照依赖和规则打包成符合生产环境部署的前端资源。还可以将按需加载的模块进行代码分隔，到实际需要的时候再异步加载。通过 loader 的转换，任何形式的资源都可以视作模块，比如CommonJs模块、 AMD 模块、 ES6 模块、CSS、图片、 JSON、Coffeescript、 LES">
<meta name="keywords" content="webpack">
<meta property="og:type" content="article">
<meta property="og:title" content="webpack学习总结(二)">
<meta property="og:url" content="http://blog.poetries.top/2017/01/14/webpack-summary/index.html">
<meta property="og:site_name" content="Poetry&#39;s Blog">
<meta property="og:description" content="声明 本教程整理于互联网   webpack是当下最热门的前端资源模块化管理和打包工具。它可以将许多松散的模块按照依赖和规则打包成符合生产环境部署的前端资源。还可以将按需加载的模块进行代码分隔，到实际需要的时候再异步加载。通过 loader 的转换，任何形式的资源都可以视作模块，比如CommonJs模块、 AMD 模块、 ES6 模块、CSS、图片、 JSON、Coffeescript、 LES">
<meta property="og:locale" content="zh-Hans">
<meta property="og:image" content="https://poetries1.gitee.io/img-repo/2019/10/661.png">
<meta property="og:updated_time" content="2020-08-15T04:25:31.942Z">
<meta name="twitter:card" content="summary">
<meta name="twitter:title" content="webpack学习总结(二)">
<meta name="twitter:description" content="声明 本教程整理于互联网   webpack是当下最热门的前端资源模块化管理和打包工具。它可以将许多松散的模块按照依赖和规则打包成符合生产环境部署的前端资源。还可以将按需加载的模块进行代码分隔，到实际需要的时候再异步加载。通过 loader 的转换，任何形式的资源都可以视作模块，比如CommonJs模块、 AMD 模块、 ES6 模块、CSS、图片、 JSON、Coffeescript、 LES">
<meta name="twitter:image" content="https://poetries1.gitee.io/img-repo/2019/10/661.png">

  

  
    <link rel="icon" href="/favicon.ico">
  

  <link href="/css/styles.css?v=c114cbeddx" rel="stylesheet">
<link href="/css/other.css?v=c114cbeddx" rel="stylesheet">


  
    <link rel="stylesheet" href="/css/personal-style.css">
  

  

  
  <script type="text/javascript">
    var _hmt = _hmt || [];
    (function() {
      var hm = document.createElement("script");
      hm.src = "//hm.baidu.com/hm.js?40b1f89aa80f2527b3db779c6898c879";
      var s = document.getElementsByTagName("script")[0];
      s.parentNode.insertBefore(hm, s);
    })();
  </script>


  
  <script type="text/javascript">
	(function(){
	    var bp = document.createElement('script');
	    var curProtocol = window.location.protocol.split(':')[0];
	    if (curProtocol === 'https') {
	        bp.src = 'https://zz.bdstatic.com/linksubmit/push.js';        
	    }
	    else {
	        bp.src = 'http://push.zhanzhang.baidu.com/push.js';
	    }
	    var s = document.getElementsByTagName("script")[0];
	    s.parentNode.insertBefore(bp, s);
	})();
  </script>



  
    <script async src="https://busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js"></script>
    <link rel="stylesheet" href="//cdn.bootcss.com/font-awesome/4.3.0/css/font-awesome.min.css">
  

  <!-- 聊天系统 -->
  
    
   <link type="text/css" rel="stylesheet" href="/renxi/default.css">
   <style>
      #modal {
        position: static !important;
      }
      .filter {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
        background: #fe5757;
        animation: colorChange 30s ease-in-out infinite;
        animation-fill-mode: both;
        mix-blend-mode: overlay;
      }
  
      @keyframes colorChange {
        0%, 100% {
            opacity: 0;
        }
        50% {
            opacity: .9;
        }
      }
   </style>
</head>
</html>
<body>
  
  
    <span id="toolbox-mobile" class="toolbox-mobile">导航</span>
  

  <div class="post-header CENTER">
   
  <div class="toolbox">
    <a class="toolbox-entry" href="/">
      <span class="toolbox-entry-text">导航</span>
      <i class="icon-angle-down"></i>
      <i class="icon-home"></i>
    </a>
    <ul class="list-toolbox">
      
        <li class="item-toolbox">
          <a
            class="CIRCLE"
            href="/archives/"
            rel="noopener noreferrer"
            target="_self"
            >
            博客
          </a>
        </li>
      
        <li class="item-toolbox">
          <a
            class="CIRCLE"
            href="/categories/"
            rel="noopener noreferrer"
            target="_self"
            >
            分类
          </a>
        </li>
      
        <li class="item-toolbox">
          <a
            class="CIRCLE"
            href="/tags/"
            rel="noopener noreferrer"
            target="_self"
            >
            标签
          </a>
        </li>
      
        <li class="item-toolbox">
          <a
            class="CIRCLE"
            href="/search/"
            rel="noopener noreferrer"
            target="_self"
            >
            搜索
          </a>
        </li>
      
        <li class="item-toolbox">
          <a
            class="CIRCLE"
            href="/link/"
            rel="noopener noreferrer"
            target="_self"
            >
            友链
          </a>
        </li>
      
        <li class="item-toolbox">
          <a
            class="CIRCLE"
            href="/about/"
            rel="noopener noreferrer"
            target="_self"
            >
            关于
          </a>
        </li>
      
    </ul>
  </div>


</div>


  <div id="toc" class="toc-article">
    <strong class="toc-title">文章目录<i class="iconfont toc-title" style="display:inline-block;color:#87998d;width:20px;height:20px;">&#xf004b;</i></strong>
    <ol class="toc"><li class="toc-item toc-level-2"><a class="toc-link" href="#一、模块系统的演进"><span class="toc-text">一、模块系统的演进</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#1-1-CommonJS"><span class="toc-text">1.1 CommonJS</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-2-AMD"><span class="toc-text">1.2 AMD</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-3-CMD"><span class="toc-text">1.3 CMD</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-4-ES6-模块"><span class="toc-text">1.4 ES6 模块</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-5-前端模块加载"><span class="toc-text">1.5 前端模块加载</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-6-所有资源都是模块"><span class="toc-text">1.6 所有资源都是模块</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-7-静态分析"><span class="toc-text">1.7 静态分析</span></a></li></ol></li><li class="toc-item toc-level-2"><a class="toc-link" href="#二、webpack基础知识"><span class="toc-text">二、webpack基础知识</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#2-1-什么是-Webpack"><span class="toc-text">2.1 什么是 Webpack</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#2-2-Webpack-的特点"><span class="toc-text">2.2  Webpack 的特点</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#2-3-总览"><span class="toc-text">2.3 总览</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#2-4-安装"><span class="toc-text">2.4 安装</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#2-5-建立一个项目"><span class="toc-text">2.5  建立一个项目</span></a></li></ol></li><li class="toc-item toc-level-2"><a class="toc-link" href="#三、Loader"><span class="toc-text">三、Loader</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#3-1-loader-有哪些特性"><span class="toc-text">3.1  loader 有哪些特性</span></a></li></ol></li><li class="toc-item toc-level-2"><a class="toc-link" href="#四、配置webpack"><span class="toc-text">四、配置webpack</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#4-1-配置webpack-dev-server"><span class="toc-text">4.1 配置webpack-dev-server</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#4-2-添加CSS样式"><span class="toc-text">4.2 添加CSS样式</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#4-3-处理图片和其他静态文件"><span class="toc-text">4.3 处理图片和其他静态文件</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#4-4-添加第三方库"><span class="toc-text">4.4 添加第三方库</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#4-5-添加ES6的支持"><span class="toc-text">4.5 添加ES6的支持</span></a></li></ol></li><li class="toc-item toc-level-2"><a class="toc-link" href="#五、插件"><span class="toc-text">五、插件</span></a></li><li class="toc-item toc-level-2"><a class="toc-link" href="#六、开发环境"><span class="toc-text">六、开发环境</span></a></li><li class="toc-item toc-level-2"><a class="toc-link" href="#七、故障处理"><span class="toc-text">七、故障处理</span></a></li><li class="toc-item toc-level-2"><a class="toc-link" href="#八、开发和部署技巧"><span class="toc-text">八、开发和部署技巧</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#8-1-启用source-map"><span class="toc-text">8.1 启用source-map</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#8-2-使用preLoaders和postLoaders"><span class="toc-text">8.2 使用preLoaders和postLoaders</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#8-3-部署上线"><span class="toc-text">8.3 部署上线</span></a></li></ol></li></ol>
  </div>
  




<div class="content content-post CENTER">
   <!-- canvas 彩带 -->
<canvas id="evanyou" width="1302" height="678" style="position: fixed;width: 100%;height: 100%;top: 0;left:0;z-index:-1;"></canvas>

<article id="post-webpack-summary" class="article article-type-post" itemprop="blogPost">
  <header class="article-header" style="position:relative;">
    <h1 class="post-title">webpack学习总结(二)</h1>

    <div class="article-meta">
      <span>
        <i class="icon-calendar"></i>
        <span>2017.01.14</span>
      </span>

      
        <span class="article-author">
          <i class="icon-user"></i>
          <span>Poetry</span>
        </span>
      

      
  <span class="article-category">
    <i class="icon-list"></i>
    <a class="article-category-link" href="/categories/Build/">Build</a>
  </span>



      

      
      <i class="fa fa-eye"></i> 
        <span id="busuanzi_container_page_pv">
           &nbsp热度 <span id="busuanzi_value_page_pv">
           <i class="fa fa-spinner fa-spin"></i></span>℃
        </span>
      
      
       
          <span class="post-count">
            <i class="fa fa-file-word-o"></i>&nbsp
            <span>字数统计 6.7k字</span>
          </span>

          <span class="post-count">
            <i class="fa fa-columns"></i>&nbsp
            <span>阅读时长 27分</span>
          </span>
      
      
    </div>

    <i class="iconfont" id="toc-eye" style="display:inline-block;color:#b36619;position:absolute;top:0;right:0;cursor:pointer;
    font-size: 24px;">&#xe61c;</i>

  </header>

  <div class="article-content">
    
      <div id="container">
        <blockquote>
<p>声明 本教程整理于互联网</p>
</blockquote>
<blockquote>
<p><code>webpack</code>是当下最热门的前端资源模块化管理和打包工具。它可以将许多松散的模块按照依赖和规则打包成符合生产环境部署的前端资源。还可以将按需加载的模块进行代码分隔，到实际需要的时候再异步加载。通过 <code>loader</code> 的转换，任何形式的资源都可以视作模块，比如<code>CommonJs</code>模块、 <code>AMD</code> 模块、 <code>ES6</code> 模块、<code>CSS</code>、图片、 <code>JSON</code>、<code>Coffeescript</code>、 LESS等</p>
</blockquote>
<h2 id="一、模块系统的演进"><a href="#一、模块系统的演进" class="headerlink" title="一、模块系统的演进"></a>一、模块系统的演进</h2><hr>
<ul>
<li><p>模块系统主要解决模块的定义、依赖和导出，先来看看已经存在的模块系统</p>
</li>
<li><p><code>&lt;script&gt;</code>标签</p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">&lt;script src=&quot;module1.js&quot;&gt;&lt;/script&gt;</span><br><span class="line">&lt;script src=&quot;module2.js&quot;&gt;&lt;/script&gt;</span><br><span class="line">&lt;script src=&quot;libraryA.js&quot;&gt;&lt;/script&gt;</span><br><span class="line">&lt;script src=&quot;module3.js&quot;&gt;&lt;/script&gt;</span><br></pre></td></tr></table></figure>
<ul>
<li><p>这是最原始的 <code>JavaScript</code> 文件加载方式，如果把每一个文件看做是一个模块，那么他们的接口通常是暴露在全局作用域下，也就是定义在 <code>window</code> 对象中，不同模块的接口调用都是一个作用域中，一些复杂的框架，会使用命名空间的概念来组织这些模块的接口，典型的例子如 <code>YUI</code> 库</p>
</li>
<li><p>这种原始的加载方式暴露了一些显而易见的弊端</p>
<ul>
<li>全局作用域下容易造成变量冲突</li>
<li>文件只能按照 <code>&lt;script&gt;</code>的书写顺序进行加载</li>
<li>开发人员必须主观解决模块和代码库的依赖关系</li>
<li>在大型项目中各种资源难以管理，长期积累的问题导致代码库混乱不堪</li>
</ul>
</li>
</ul>
<h3 id="1-1-CommonJS"><a href="#1-1-CommonJS" class="headerlink" title="1.1 CommonJS"></a>1.1 CommonJS</h3><hr>
<ul>
<li>服务器端的 <code>Node.js</code>遵循 <code>CommonJS</code>规范，该规范的核心思想是允许模块通过<code>require</code>方法来同步加载所要依赖的其他模块，然后通过 <code>exports</code> 或 <code>module.exports</code> 来导出需要暴露的接口</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">require(&quot;module&quot;);</span><br><span class="line">require(&quot;../file.js&quot;);</span><br><span class="line">exports.doStuff = function() &#123;&#125;;</span><br><span class="line">module.exports = someValue;</span><br></pre></td></tr></table></figure>
<ul>
<li>优点：<ul>
<li>服务器端模块便于重用</li>
<li><code>NPM</code> 中已经有将近<code>20</code>万个可以使用模块包</li>
<li>简单并容易使用</li>
</ul>
</li>
<li><p>缺点：</p>
<ul>
<li>同步的模块加载方式不适合在浏览器环境中，同步意味着阻塞加载，浏览器资源是异步加载的</li>
<li>不能非阻塞的并行加载多个模块</li>
</ul>
</li>
<li><p>实现</p>
<ul>
<li>服务器端的 <code>Node.js</code></li>
<li><code>Browserify</code>，浏览器端的 <code>CommonJS</code> 实现，可以使用 <code>NPM</code>的模块，但是编译打包后的文件体积可能很大</li>
<li><code>modules-webmake</code>，类似<code>Browserify</code>，还不如 <code>Browserify</code> 灵活</li>
<li><code>wreq</code>，<code>Browserify</code>的前身</li>
</ul>
</li>
</ul>
<h3 id="1-2-AMD"><a href="#1-2-AMD" class="headerlink" title="1.2 AMD"></a>1.2 AMD</h3><hr>
<blockquote>
<p><code>Asynchronous Module Definition</code> 规范其实只有一个主要接口<code>define(id?, dependencies?,factory)</code> ，它要在声明模块的时候指定所有的依赖 <code>dependencies</code>，并且还要当做形参传到<code>factory</code> 中，对于依赖的模块提前执行，依赖前置</p>
</blockquote>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">define(&quot;module&quot;, [&quot;dep1&quot;, &quot;dep2&quot;], function(d1, d2) &#123;</span><br><span class="line">return someExportedValue;</span><br><span class="line">&#125;);require([&quot;module&quot;, &quot;../file&quot;], function(module, file) &#123; /* ... */ &#125;);</span><br></pre></td></tr></table></figure>
<ul>
<li><p>优点：</p>
<ul>
<li>适合在浏览器环境中异步加载模块</li>
<li>可以并行加载多个模块</li>
</ul>
</li>
<li><p>缺点：</p>
<ul>
<li>提高了开发成本，代码的阅读和书写比较困难，模块定义方式的语义 不顺畅</li>
<li>不符合通用的模块化思维方式，是一种妥协的实现</li>
</ul>
</li>
<li><p>实现：</p>
<ul>
<li><code>RequireJS</code></li>
<li><code>curl</code></li>
</ul>
</li>
</ul>
<h3 id="1-3-CMD"><a href="#1-3-CMD" class="headerlink" title="1.3 CMD"></a>1.3 CMD</h3><hr>
<ul>
<li><code>Common Module Definition</code> 规范和 <code>AMD</code>很相似，尽量保持简单，并与 <code>CommonJS</code> 和<code>Node.js</code> 的 <code>Modules</code>规范保持了很大的兼容性</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">define(function(require, exports, module) &#123;</span><br><span class="line">var $ = require(&apos;jquery&apos;);</span><br><span class="line">var Spinning = require(&apos;./spinning&apos;);</span><br><span class="line">exports.doSomething = ...</span><br><span class="line">module.exports = ...</span><br><span class="line">&#125;)</span><br></pre></td></tr></table></figure>
<ul>
<li>优点：<ul>
<li>依赖就近，延迟执行</li>
<li>可以很容易在 <code>Node.js</code> 中运行</li>
</ul>
</li>
<li>缺点：<ul>
<li>依赖 <code>SPM</code> 打包，模块的加载逻辑偏重</li>
</ul>
</li>
<li>实现：<ul>
<li><code>Sea.js</code></li>
<li><code>coolie</code></li>
</ul>
</li>
</ul>
<h3 id="1-4-ES6-模块"><a href="#1-4-ES6-模块" class="headerlink" title="1.4 ES6 模块"></a>1.4 ES6 模块</h3><hr>
<ul>
<li><code>EcmaScript6</code> 标准增加了 <code>JavaScript</code>语言层面的模块体系定义。<code>ES6</code> 模块的设计思想，是尽量的静态化，使得编译时就能确定模块的依赖关系，以及输入和输出的变量。<code>CommonJS</code> 和<code>AMD</code> 模块，都只能在运行时确定这些东西</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">import &quot;jquery&quot;;</span><br><span class="line">export function doStuff() &#123;&#125;</span><br><span class="line">module &quot;localModule&quot; &#123;&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>优点：<ul>
<li>容易进行静态分析</li>
<li>面向未来的 <code>EcmaScript</code>标准</li>
</ul>
</li>
<li>缺点：<ul>
<li>原生浏览器端还没有实现该标准</li>
<li>全新的命令字，新版的 <code>Node.js</code>才支持</li>
</ul>
</li>
<li>实现：<ul>
<li><code>Babel</code></li>
</ul>
</li>
</ul>
<h3 id="1-5-前端模块加载"><a href="#1-5-前端模块加载" class="headerlink" title="1.5 前端模块加载"></a>1.5 前端模块加载</h3><hr>
<ul>
<li>前端模块要在客户端中执行，所以他们需要增量加载到浏览器中</li>
<li>模块的加载和传输，我们首先能想到两种极端的方式，一种是每个模块文件都单独请求，另一种是把所有模块打包成一个文件然后只请求一次。显而易见，每个模块都发起单独的请求造成了请求次数过多，导致应用启动速度慢；一次请求加载所有模块导致流量浪费、初始化过程慢。这两种方式都不是好的解决方案，它们过于简单粗暴</li>
<li>分块传输，按需进行懒加载，在实际用到某些模块的时候再增量更新，才是较为合理的模块加载方案</li>
<li>要实现模块的按需加载，就需要一个对整个代码库中的模块进行静态分析、编译打包的过<br>程</li>
</ul>
<h3 id="1-6-所有资源都是模块"><a href="#1-6-所有资源都是模块" class="headerlink" title="1.6 所有资源都是模块"></a>1.6 所有资源都是模块</h3><hr>
<blockquote>
<p>在上面的分析过程中，我们提到的模块仅仅是指<code>JavaScript</code>模块文件。然而，在前端开发过程中还涉及到样式、图片、字体、<code>HTML</code> 模板等等众多的资源。这些资源还会以各种方言的形式存在，比如 <code>coffeescript</code>、 <code>less</code>、 <code>sass</code>、众多的模板库、多语言系统（<code>i18n</code>）等</p>
</blockquote>
<ul>
<li>如果他们都可以视作模块，并且都可以通过 <code>require</code> 的方式来加载，将带来优雅的开发体验，比如</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">require(&quot;./style.css&quot;);</span><br><span class="line">require(&quot;./style.less&quot;);</span><br><span class="line">require(&quot;./template.jade&quot;);</span><br><span class="line">require(&quot;./image.png&quot;);</span><br></pre></td></tr></table></figure>
<ul>
<li>那么如何做到让 <code>require</code> 能加载各种资源呢？</li>
</ul>
<h3 id="1-7-静态分析"><a href="#1-7-静态分析" class="headerlink" title="1.7 静态分析"></a>1.7 静态分析</h3><hr>
<blockquote>
<p>在编译的时候，要对整个代码进行静态分析，分析出各个模块的类型和它们依赖关系，然后将不同类型的模块提交给适配的加载器来处理。比如一个用<code>LESS</code> 写的样式模块，可以先用<code>LESS</code> 加载器将它转成一个<code>CSS</code>模块，在通过 <code>CSS</code> 模块把他插入到页面的 <code>&lt;style&gt;</code> 标签中执行。<code>Webpack</code>就是在这样的需求中应运而生</p>
</blockquote>
<h2 id="二、webpack基础知识"><a href="#二、webpack基础知识" class="headerlink" title="二、webpack基础知识"></a>二、webpack基础知识</h2><hr>
<h3 id="2-1-什么是-Webpack"><a href="#2-1-什么是-Webpack" class="headerlink" title="2.1 什么是 Webpack"></a>2.1 什么是 Webpack</h3><hr>
<ul>
<li><code>Webpack</code> 是一个模块打包器。它将根据模块的依赖关系进行静态分析，然后将这些模块按照指定的规则生成对应的静态资源</li>
</ul>
<h3 id="2-2-Webpack-的特点"><a href="#2-2-Webpack-的特点" class="headerlink" title="2.2  Webpack 的特点"></a>2.2  Webpack 的特点</h3><hr>
<ul>
<li><p>代码拆分</p>
<ul>
<li><code>Webpack</code>有两种组织模块依赖的方式，同步和异步。异步依赖作为分割点，形成一个新的块。在优化了依赖树后，每一个异步区块都作为一个文件被打包</li>
</ul>
</li>
<li><p>Loader</p>
<ul>
<li><code>Webpack</code> 本身只能处理原生的 <code>JavaScript</code> 模块，但是 <code>loader</code> 转换器可以将各种类型的资源转换成 <code>JavaScript</code> 模块。这样，任何资源都可以成为 <code>Webpack</code>可以处理的模块</li>
</ul>
</li>
<li>智能解析<ul>
<li><code>Webpack</code> 有一个智能解析器，几乎可以处理任何第三方库，无论它们的模块形式是<code>CommonJS</code>、 <code>AMD</code>还是普通的 <code>JS</code> 文件。甚至在加载依赖的时候，允许使用动态表达式<code>require(&quot;./templates/&quot; + name + &quot;.jade&quot;)</code></li>
</ul>
</li>
<li>插件系统<ul>
<li><code>Webpack</code> 还有一个功能丰富的插件系统。大多数内容功能都是基于这个插件系统运行的，还可以开发和使用开源的 <code>Webpack</code> 插件，来满足各式各样的需求</li>
</ul>
</li>
<li>快速运行<ul>
<li><code>Webpack</code> 使用异步 <code>I/O</code> 和多级缓存提高运行效率，这使得 <code>Webpack</code>能够以令人难以置信的速度快速增量编译</li>
</ul>
</li>
</ul>
<h3 id="2-3-总览"><a href="#2-3-总览" class="headerlink" title="2.3 总览"></a>2.3 总览</h3><hr>
<ul>
<li>他的目的就是把有依赖关系的各种文件打包成一系列的静<br>态资源</li>
<li><code>webpack</code>简单点来说就就是一个配置文件，所有的魔力都是在这一个文件中发生的。 这个配置文件主要分<br>为三大块<ul>
<li><code>entry</code> 入口文件 让<code>webpack</code>用哪个文件作为项目的入口</li>
<li><code>output</code> 出口 让<code>webpack</code>把处理完成的文件放在哪里</li>
<li><code>module</code> 模块 要用什么不同的模块来处理各种类型的文件</li>
</ul>
</li>
</ul>
<p><img src="https://poetries1.gitee.io/img-repo/2019/10/661.png" alt></p>
<h3 id="2-4-安装"><a href="#2-4-安装" class="headerlink" title="2.4 安装"></a>2.4 安装</h3><hr>
<ul>
<li>先装好node和npm，因为webpack是一个基于node的项目。然后</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install -g webpack</span><br></pre></td></tr></table></figure>
<ul>
<li>此时 <code>Webpack</code>已经安装到了全局环境下，可以通过命令行<code>webpack -h</code> 试试</li>
<li>通常我们会将 <code>Webpack</code>安装到项目的依赖中，这样就可以使用项目本地版本的 <code>Webpack</code></li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line"># 进入项目目录</span><br><span class="line"># 确定已经有 package.json，没有就通过 npm init 创建</span><br><span class="line"># 安装 webpack 依赖</span><br><span class="line">$ npm install webpack --save-dev</span><br></pre></td></tr></table></figure>
<blockquote>
<p><code>Webpack</code> 目前有两个主版本，一个是在 master 主干的稳定版，一个是在 <code>webpack-2</code> 分支的测试版，测试版拥有一些实验性功能并且和稳定版不兼容，在正式项目中应该使用稳定版</p>
</blockquote>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line"># 查看 webpack 版本信息</span><br><span class="line">$ npm info webpack</span><br><span class="line"># 安装指定版本的 webpack</span><br><span class="line">$ npm install webpack@1.12.x --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>如果需要使用 <code>Webpack</code>开发工具，要单独安装</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$ npm install webpack-dev-server --save-dev</span><br></pre></td></tr></table></figure>
<h3 id="2-5-建立一个项目"><a href="#2-5-建立一个项目" class="headerlink" title="2.5  建立一个项目"></a>2.5  建立一个项目</h3><hr>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">mkdir webpack</span><br><span class="line">cd webpack</span><br><span class="line">npm init</span><br></pre></td></tr></table></figure>
<ul>
<li>如果你使用git管理你的这个项目的话，建议你新建一个<code>.gitignore</code>文件，不要让<code>git</code>提交一些<code>node</code>依赖的模<br>块</li>
<li>创建一个静态页面 <code>index.html</code> 和一个 <code>JS</code> 入口文件 <code>entry.js：</code></li>
</ul>
<figure class="highlight html"><table><tr><td class="code"><pre><span class="line"><span class="comment">&lt;!-- index.html --&gt;</span></span><br><span class="line"><span class="tag">&lt;<span class="name">html</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;<span class="name">head</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;<span class="name">meta</span> <span class="attr">charset</span>=<span class="string">"utf-8"</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;/<span class="name">head</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;<span class="name">body</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;<span class="name">script</span> <span class="attr">src</span>=<span class="string">"bundle.js"</span>&gt;</span><span class="tag">&lt;/<span class="name">script</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;/<span class="name">body</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;/<span class="name">html</span>&gt;</span></span><br></pre></td></tr></table></figure>
<figure class="highlight javascript"><table><tr><td class="code"><pre><span class="line"><span class="comment">// entry.js</span></span><br><span class="line"><span class="built_in">document</span>.write(<span class="string">'It works.'</span>)</span><br></pre></td></tr></table></figure>
<ul>
<li>然后编译 <code>entry.js</code> 并打包到 <code>bundle.js</code>：</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$ webpack entry.js bundle.js</span><br></pre></td></tr></table></figure>
<ul>
<li>打包过程会显示日志：</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">Hash: e964f90ec65eb2c29bb9</span><br><span class="line">Version: webpack 1.12.2</span><br><span class="line">Time: 54ms</span><br><span class="line">Asset Size Chunks Chunk Names</span><br><span class="line">bundle.js 1.42 kB 0 [emitted] main</span><br><span class="line">[0] ./entry.js 27 bytes &#123;0&#125; [built]</span><br></pre></td></tr></table></figure>
<ul>
<li>用浏览器打开 <code>index.html</code> 将会看到 <code>It works.</code></li>
<li>接下来添加一个模块 <code>module.js</code>并修改入口 <code>entry.js</code>：</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">// module.js</span><br><span class="line">module.exports = &apos;It works from module.js.&apos;</span><br></pre></td></tr></table></figure>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">// entry.js</span><br><span class="line">document.write(&apos;It works.&apos;)</span><br><span class="line">document.write(require(&apos;./module.js&apos;)) // 添加模块</span><br></pre></td></tr></table></figure>
<ul>
<li><p>重新打包 <code>webpack entry.js bundle.js</code>后刷新页面看到变化 <code>It works.It works from module.js.</code></p>
</li>
<li><p><code>Webpack</code> 会分析入口文件，解析包含依赖关系的各个文件。这些文件（模块）都打包到<code>bundle.js</code> 。<code>Webpack</code> 会给每个模块分配一个唯一的<code>id</code>并通过这个<code>id</code> 索引和访问模块。在页面启动时，会先执<code>entry.js</code> 中的代码，其它模块会在运行<code>require</code>的时候再执行</p>
</li>
</ul>
<h2 id="三、Loader"><a href="#三、Loader" class="headerlink" title="三、Loader"></a>三、Loader</h2><hr>
<ul>
<li><code>Webpack</code> 本身只能处理 <code>JavaScript</code> 模块，如果要处理其他类型的文件，就需要使用 <code>loader</code>进行转换</li>
<li><code>Loader</code> 可以理解为是模块和资源的转换器，它本身是一个函数，接受源文件作为参数，返回转换的结果。这样，我们就可以通过 require 来加载任何类型的模块或文件，比如CoffeeScript、 JSX、 LESS </li>
</ul>
<h3 id="3-1-loader-有哪些特性"><a href="#3-1-loader-有哪些特性" class="headerlink" title="3.1  loader 有哪些特性"></a>3.1  loader 有哪些特性</h3><hr>
<ul>
<li><code>Loader</code> 可以通过管道方式链式调用，每个 <code>loader</code> 可以把资源转换成任意格式并传递给下一个 <code>loader</code> ，但是最后一个 <code>loader</code> 必须返回 <code>JavaScript</code></li>
<li><code>Loader</code>可以同步或异步执行。</li>
<li><code>Loader</code> 运行在 <code>node.js</code> 环境中，所以可以做任何可能的事情。</li>
<li><code>Loader</code> 可以接受参数，以此来传递配置项给 <code>loader</code>。</li>
<li><code>Loader</code> 可以通过文件扩展名（或正则表达式）绑定给不同类型的文件。</li>
<li><code>Loader</code>可以通过 <code>npm</code>发布和安装。</li>
<li>除了通过 <code>package.json</code>的 <code>main</code>指定，通常的模块也可以导出一个 <code>loader</code>来使用。</li>
<li><code>Loader</code> 可以访问配置。</li>
<li>插件可以让 <code>loader</code> 拥有更多特性。</li>
<li><code>Loader</code>可以分发出附加的任意文件</li>
</ul>
<blockquote>
<ul>
<li><code>Loader</code>本身也是运行在 <code>node.js</code> 环境中的 <code>JavaScript</code>模块，它通常会返回一个函数。大多数情况下，我们通过 npm 来管理 <code>loader</code>，但是你也可以在项目中自己写 <code>loader</code> 模块</li>
<li>按照惯例，而非必须，<code>loader</code> 一般以 <code>xxx-loader</code> 的方式命名， <code>xxx</code> 代表了这个 <code>loader</code>要做的转换功能，比如 <code>json-loader</code></li>
<li>在引用 <code>loader</code> 的时候可以使用全名 <code>json-loader</code> ，或者使用短名 <code>json</code>。这个命名规则和搜索优先级顺序在<code>webpack</code> 的<code>resolveLoader.moduleTemplates api</code>中定义</li>
</ul>
</blockquote>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">Default: [&quot;*-webpack-loader&quot;, &quot;*-web-loader&quot;, &quot;*-loader&quot;, &quot;*&quot;]</span><br></pre></td></tr></table></figure>
<ul>
<li><code>Loader</code>可以在 <code>require()</code>引用模块的时候添加，也可以在 <code>webpack</code> 全局配置中进行绑定，还可以通过命令行的方式使用</li>
<li>我们要在页面中引入一个<code>CSS</code>文件<code>style.css</code>，首页将 <code>style.css</code>也看成是一个模块，然后用 <code>css-loader</code> 来读取它，再用<code>style-loader</code>把它插入到页面中</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">/* style.css */</span><br><span class="line">body &#123; </span><br><span class="line">  background: yellow; </span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>修改 entry.js：</p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">require(&quot;!style!css!./style.css&quot;) // 载入 style.css</span><br><span class="line">document.write(&apos;It works.&apos;)</span><br><span class="line">document.write(require(&apos;./module.js&apos;))</span><br></pre></td></tr></table></figure>
<p>安装 loader：</p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install css-loader style-loader</span><br></pre></td></tr></table></figure>
<p>重新编译打包，刷新页面，就可以看到黄色的页面背景了</p>
<blockquote>
<p>如果每次   require CSS   文件的时候都要写   loader   前缀，是一件很繁琐的事情。我们可以根据模块类型（扩展名）来自动绑定需要的   loader  </p>
</blockquote>
<ul>
<li>将   <code>entry.js</code>  中的 <code>require(&quot;!style!css!./style.css&quot;)</code>修改为 <code>require(&quot;./style.css&quot;)</code>，然后执行：</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$ webpack entry.js bundle.js --module-bind &apos;css=style!css&apos;</span><br><span class="line"># 有些环境下可能需要使用双引号</span><br><span class="line">$ webpack entry.js bundle.js --module-bind &quot;css=style!css&quot;</span><br></pre></td></tr></table></figure>
<ul>
<li>显然，这两种使用 <code>loader</code> 的方式，效果是一样的</li>
</ul>
<h2 id="四、配置webpack"><a href="#四、配置webpack" class="headerlink" title="四、配置webpack"></a>四、配置webpack</h2><hr>
<blockquote>
<p><code>Webpack</code>在执行的时候，除了在命令行传入参数，还可以通过指定的配置文件来执行。默认情况下，会搜索当前目<code>webpack.config.js</code> 文件，这个文件是一个 <code>node.js</code>模块，返回一个<code>json</code> 格式的配置信息对象，或者通过 <code>--config</code> 选项来指定配置文件</p>
</blockquote>
<ul>
<li><p>现在开始配置<code>webpack</code>，目标是把这两个<code>js</code>文件合并成一个文件. 我们可以自己在<code>build</code>文件夹里面手动建一个<code>index.html</code>文件夹，然后再把合并以后的<code>js</code>引用在里面，但是这样有些麻烦，所以我们这里安装一个<code>plugin</code>，可以自动快速的帮我们生成<code>HTML</code></p>
</li>
<li><p>创建一个配置文件 <code>webpack.config.js</code></p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">var webpack = require(&apos;webpack&apos;)</span><br><span class="line">module.exports = &#123;</span><br><span class="line">entry: &apos;./entry.js&apos;,</span><br><span class="line">output: &#123;</span><br><span class="line">path: __dirname,</span><br><span class="line">filename: &apos;bundle.js&apos;</span><br><span class="line">&#125;,</span><br><span class="line">module: &#123;</span><br><span class="line">loaders: [</span><br><span class="line">&#123;test: /\.css$/, loader: &apos;style!css&apos;&#125;</span><br><span class="line">]</span><br><span class="line">&#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>同时简化 <code>entry.js</code> 中的 <code>style.css</code> 加载方式</li>
<li><code>require(&#39;./style.css&#39;)</code></li>
<li>最后运行 <code>webpack</code> ，可以看到 <code>webpack</code> 通过配置文件执行的结果和通过命令行<code>webpack entry.js bundle.js --module-bind &#39;css=style!css&#39;</code>执行的结果是一样的</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install html-webpack-plugin --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>有了这个插件 开始写<code>config</code>文件</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">var path = require(&apos;path&apos;);</span><br><span class="line">var HtmlwebpackPlugin = require(&apos;html-webpack-plugin&apos;);</span><br><span class="line">//定义了一些文件夹的路径</span><br><span class="line">var ROOT_PATH = path.resolve(__dirname);</span><br><span class="line">var APP_PATH = path.resolve(ROOT_PATH, &apos;app&apos;);</span><br><span class="line">var BUILD_PATH = path.resolve(ROOT_PATH, &apos;build&apos;);</span><br><span class="line">module.exports = &#123;</span><br><span class="line">//项目的文件夹 可以直接用文件夹名称 默认会找index.js 也可以确定是哪个文件名字</span><br><span class="line">entry: APP_PATH,</span><br><span class="line">//输出的文件名 合并以后的js会命名为bundle.js</span><br><span class="line">output: &#123;</span><br><span class="line">path: BUILD_PATH,</span><br><span class="line">filename: &apos;bundle.js&apos;</span><br><span class="line">&#125;,</span><br><span class="line">//添加我们的插件 会自动生成一个html文件</span><br><span class="line">plugins: [</span><br><span class="line">new HtmlwebpackPlugin(&#123;</span><br><span class="line">title: &apos;Hello World app&apos;</span><br><span class="line">&#125;)</span><br><span class="line">]</span><br><span class="line">&#125;;</span><br></pre></td></tr></table></figure>
<ul>
<li>然后在项目根目录运行</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">webpack</span><br></pre></td></tr></table></figure>
<ul>
<li>你会发现多出来一个build文件夹，直接点开里面的html文件，你会发现我们可爱的<code>“hello world”</code>已经插入到页面了。我们的任务完成了，成功生成<code>html</code>，合并<code>js</code>，<code>html</code>引入了<code>js</code>，<code>js</code>被执行了</li>
</ul>
<h3 id="4-1-配置webpack-dev-server"><a href="#4-1-配置webpack-dev-server" class="headerlink" title="4.1 配置webpack-dev-server"></a>4.1 配置webpack-dev-server</h3><hr>
<ul>
<li><p>上面任务虽然完成了，但是我们要不断运行程序然后查看页面，所以最好新建一个开发服务器，可以<code>serve</code>我们<code>pack</code>以后的代码，并且当代码更新的时候自动刷新浏览器</p>
</li>
<li><p>安装<code>webpack-dev-server</code></p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install webpack-dev-server --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>安装完毕后在<code>config</code>中添加配置</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">module.exports = &#123;</span><br><span class="line"></span><br><span class="line">devServer: &#123;</span><br><span class="line">historyApiFallback: true,</span><br><span class="line">hot: true,</span><br><span class="line">inline: true,</span><br><span class="line">progress: true,</span><br><span class="line">&#125;,</span><br><span class="line"></span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>然后再<code>package.json</code>里面配置一下运行的命令,<code>npm</code>支持自定义一些命令</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">&quot;scripts&quot;: &#123;</span><br><span class="line">&quot;start&quot;: &quot;webpack-dev-server --hot --inline&quot;</span><br><span class="line">&#125;,</span><br></pre></td></tr></table></figure>
<ul>
<li><p>在项目根目录下输入<code>npm start</code>,一堆花花绿绿的信息后server已经起来了，在浏览器里面输入<code>http://localhost:8080</code>发现<code>hello world</code>出现了，在<code>js</code>里面随便修改一些输出,然后保存,浏览器自动刷新，新的结果出现了</p>
</li>
<li><p>拓展阅读 如果你的服务器端使用的是<code>express</code>框架，你还可以直接安装<code>express</code>的<code>middleware</code>，<code>webpack</code>配合<code>express</code>，很好用</p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install webpack-dev-middleware --save-dev</span><br></pre></td></tr></table></figure>
<h3 id="4-2-添加CSS样式"><a href="#4-2-添加CSS样式" class="headerlink" title="4.2 添加CSS样式"></a>4.2 添加CSS样式</h3><ul>
<li><p>现在来添加一些样式，<code>webpack</code>使用<code>loader</code>的方式来处理各种各样的资源，比如说样式文件，我们需要两种<code>loader</code>，<code>css-loader</code>和 <code>style－loader</code>，<code>css-loader</code>会遍历<code>css</code>文件，找到所有的<code>url(...)</code>并且处理。<code>style-loader</code>会把所有的样式插入到你页面的一个<code>style tag</code>中</p>
</li>
<li><p>安装我们的<code>loader</code></p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install css-loader style-loader --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>配置<code>loader</code>，在<code>webpack.config.js</code>中</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">devServer: &#123;</span><br><span class="line">historyApiFallback: true,</span><br><span class="line">hot: true,</span><br><span class="line">inline: true,</span><br><span class="line">progress: true,</span><br><span class="line">&#125;,</span><br><span class="line">...</span><br><span class="line">module: &#123;</span><br><span class="line">loaders: [</span><br><span class="line">&#123;</span><br><span class="line">test: /\.css$/,</span><br><span class="line">loaders: [&apos;style&apos;, &apos;css&apos;],</span><br><span class="line">include: APP_PATH</span><br><span class="line">&#125;</span><br><span class="line">]</span><br><span class="line">&#125;,</span><br><span class="line">...</span><br><span class="line">plugins: [</span><br><span class="line">new HtmlwebpackPlugin(&#123;</span><br><span class="line">title: &apos;Hello World app&apos;</span><br></pre></td></tr></table></figure>
<ul>
<li><p>看<code>loaders</code>的书写方式，test里面包含一个正则，包含需要匹配的文件，<code>loaders</code>是一个数组，包含要处理这些程序的<code>loaders</code>，这里我们用了<code>css</code>和<code>style</code>，注意<code>loaders</code>的处理顺序是从右到左的，这里就是先运行<code>css-loader</code>然后是<code>style-loader</code></p>
</li>
<li><p>新建一个样式文件 <code>main.css</code></p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">h1 &#123;</span><br><span class="line">color: red;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>记得在入口文件<code>index.js</code>中引用</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">require(&apos;./main.css&apos;);</span><br></pre></td></tr></table></figure>
<ul>
<li><p>然后发现标题变成红色的了，<code>webpack</code>的理念是基于项目处理的，把对应的文件格式给对应的<code>loader</code>处理，然后你就不用管了，它会决定怎么压缩，编译</p>
</li>
<li><p>那现在想使用一些有爱的<code>css</code>预编译程序，来点<code>sass</code>吧。 你可能已经想到了，再来个loader就行啦，确实是<br>这样简单</p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install sass-loader --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>稍微修改一下<code>config</code>，删掉我们先前添加的<code>css</code>规则，加上下面的<code>loader</code></li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">&#123;</span><br><span class="line">test: /\.scss$/,</span><br><span class="line">loaders: [&apos;style&apos;, &apos;css&apos;, &apos;sass&apos;],</span><br><span class="line">include: APP_PATH</span><br><span class="line">&#125;,</span><br></pre></td></tr></table></figure>
<ul>
<li>添加两个<code>sass</code>文件，<code>variables.scss</code>和<code>main.scss</code></li>
</ul>
<p><code>variables.scss</code></p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$red: red;</span><br></pre></td></tr></table></figure>
<p><code>main.scss</code></p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">@import &quot;./variables.scss&quot;;</span><br><span class="line">h1 &#123;</span><br><span class="line">color: $red;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>在<code>index.js</code>中引用<br><figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">require(&apos;./main.scss&apos;);</span><br></pre></td></tr></table></figure></p>
<ul>
<li>然后发现标题如愿变红</li>
</ul>
<h3 id="4-3-处理图片和其他静态文件"><a href="#4-3-处理图片和其他静态文件" class="headerlink" title="4.3 处理图片和其他静态文件"></a>4.3 处理图片和其他静态文件</h3><hr>
<ul>
<li><p>这个和其他一样，也许你也已经会玩了。安装<code>loader</code>，处理文件。诸如图片，字体等等，不过有个神奇的地方它可以根据你的需求将一些图片自动转成<code>base64</code>编码的，为你减轻很多的网络请求</p>
</li>
<li><p>安装<code>url-loader</code></p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install url-loader --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>配置<code>config</code>文件</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">&#123;</span><br><span class="line">test: /\.(png|jpg)$/,</span><br><span class="line">loader: &apos;url?limit=40000&apos;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>注意后面那个limit的参数，当你图片大小小于这个限制的时候，会自动启用base64编码图片</li>
</ul>
<p>新建一个<code>imgs</code>文件夹，往里面添加一张照片。在<code>scss</code>文件中添加如下的东西</p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">@import &quot;./variables.scss&quot;;</span><br><span class="line">h1 &#123;</span><br><span class="line">color: $red;</span><br><span class="line">background: url(&apos;./imgs/avatar.jpg&apos;);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>npm start, 然后查看图片的url</li>
</ul>
<h3 id="4-4-添加第三方库"><a href="#4-4-添加第三方库" class="headerlink" title="4.4 添加第三方库"></a>4.4 添加第三方库</h3><hr>
<ul>
<li>有的时候还想来点<code>jquery</code>，<code>moment</code>，<code>undersocre</code>之类的库<code>webpack</code>可以非常容易的做到这一点</li>
<li>那么我们现在安装在我们的<code>app</code>中添加<code>jquery</code>和<code>moment</code>的支持</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install jquery moment --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>在js中引用</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">var sub = require(&apos;./sub&apos;);</span><br><span class="line">var $ = require(&apos;jquery&apos;);</span><br><span class="line">var moment = require(&apos;moment&apos;);</span><br><span class="line">var app = document.createElement(&apos;div&apos;);</span><br><span class="line">app.innerHTML = &apos;&lt;h1&gt;Hello World it&lt;/h1&gt;&apos;;</span><br><span class="line">document.body.appendChild(app);</span><br><span class="line">app.appendChild(sub());</span><br><span class="line">$(&apos;body&apos;).append(&apos;&lt;p&gt;look at me! now is &apos; + moment().format() + &apos;&lt;/p&gt;&apos;);</span><br></pre></td></tr></table></figure>
<ul>
<li>看看浏览器，成功！ <code>jquery</code>和<code>moment</code>现在都起作用了</li>
</ul>
<h3 id="4-5-添加ES6的支持"><a href="#4-5-添加ES6的支持" class="headerlink" title="4.5 添加ES6的支持"></a>4.5 添加ES6的支持</h3><hr>
<ul>
<li>首先 装各种<code>loader</code></li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install babel-loader babel-preset-es2015 --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>配置我们的<code>config</code>文件</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">&#123;</span><br><span class="line">test: /\.jsx?$/,</span><br><span class="line">loader: &apos;babel&apos;,</span><br><span class="line">include: APP_PATH,</span><br><span class="line">query: &#123;</span><br><span class="line">presets: [&apos;es2015&apos;]</span><br><span class="line">&#125;</span><br><span class="line">&#125;,</span><br></pre></td></tr></table></figure>
<ul>
<li><p><code>es2015</code>这个参数是<code>babel</code>的<code>plugin</code>，可以支持各种最新的<code>es6</code>的特性，具体的情况看这个链接</p>
</li>
<li><p>现在我们可以改掉<code>CommonJS</code>风格的文件了</p>
</li>
</ul>
<p><code>sub.js</code></p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">export default function() &#123;</span><br><span class="line">  var element = document.createElement(&apos;h2&apos;);</span><br><span class="line">  element.innerHTML = &quot;Hello h2 world hahaha&quot;;</span><br><span class="line">  return element;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p><code>index.js</code></p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">import &apos;./main.scss&apos;;</span><br><span class="line">import generateText from &apos;./sub&apos;;</span><br><span class="line">import $ from &apos;jquery&apos;;</span><br><span class="line">import moment from &apos;moment&apos;;</span><br><span class="line">let app = document.createElement(&apos;div&apos;);</span><br><span class="line">const myPromise = Promise.resolve(42);</span><br><span class="line">myPromise.then((number) =&gt; &#123;</span><br><span class="line">$(&apos;body&apos;).append(&apos;&lt;p&gt;promise result is &apos; + number + &apos; now is &apos; + moment().format() + &apos;&lt;/p&gt;&apos;);</span><br><span class="line">&#125;);</span><br><span class="line">app.innerHTML = &apos;&lt;h1&gt;Hello World it&lt;/h1&gt;&apos;;</span><br><span class="line">document.body.appendChild(app);</span><br><span class="line">app.appendChild(generateText());</span><br></pre></td></tr></table></figure>
<ul>
<li>我们上面测试了<code>import</code>,<code>export</code>，<code>const</code>，<code>let</code>，<code>promise</code>等一系列<code>es6</code>的特性</li>
</ul>
<h2 id="五、插件"><a href="#五、插件" class="headerlink" title="五、插件"></a>五、插件</h2><hr>
<blockquote>
<ul>
<li>插件可以完成更多 <code>loader</code> 不能完成的功能</li>
<li>插件的使用一般是在 <code>webpack</code> 的配置信息 <code>plugins</code> 选项中指定</li>
<li><code>Webpack</code> 本身内置了一些常用的插件，还可以通过 <code>npm</code> 安装第三方插件</li>
<li>接下来，我们利用一个最简单的 <code>BannerPlugin</code> 内置插件来实践插件的配置和运行，这个插件的作用是给输出的文件头部添加注释信息</li>
</ul>
</blockquote>
<p>修改 <code>webpack.config.js</code>，添加 <code>plugins</code></p>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">var webpack = require(&apos;webpack&apos;)</span><br><span class="line">module.exports = &#123;</span><br><span class="line">entry: &apos;./entry.js&apos;,</span><br><span class="line">output: &#123;</span><br><span class="line">path: __dirname,</span><br><span class="line">filename: &apos;bundle.js&apos;</span><br><span class="line">&#125;,</span><br><span class="line">module: &#123;</span><br><span class="line">loaders: [</span><br><span class="line">&#123;test: /\.css$/, loader: &apos;style!css&apos;&#125;</span><br><span class="line">]</span><br><span class="line">&#125;,</span><br><span class="line">plugins: [</span><br><span class="line">new webpack.BannerPlugin(&apos;This file is created by zhaoda&apos;)</span><br><span class="line">]</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<ul>
<li>然后运行 <code>webpack</code> ，打开 <code>bundle.js</code> ，可以看到文件头部出现了我们指定的注释信息</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">/*! This file is created by zhaoda */</span><br><span class="line">/******/ (function(modules) &#123; // webpackBootstrap</span><br><span class="line">/******/ // The module cache</span><br><span class="line">/******/ var installedModules = &#123;&#125;;</span><br><span class="line">// 后面代码省略</span><br></pre></td></tr></table></figure>
<h2 id="六、开发环境"><a href="#六、开发环境" class="headerlink" title="六、开发环境"></a>六、开发环境</h2><hr>
<ul>
<li>当项目逐渐变大，<code>webpack</code> 的编译时间会变长，可以通过参数让编译的输出内容带有进度和颜色</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$ webpack --progress --colors</span><br></pre></td></tr></table></figure>
<ul>
<li>如果不想每次修改模块后都重新编译，那么可以启动监听模式。开启监听模式后，没有变化的模块会在编译后缓存到内存中，而不会每次都被重新编译，所以监听模式的整体速度是很快的</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$ webpack --progress --colors --watch</span><br></pre></td></tr></table></figure>
<blockquote>
<p>当然，使用 <code>webpack-dev-server</code>开发服务是一个更好的选择。它将在 localhost:8080 启动一个 <code>express</code> 静态资源 <code>web</code>服务器，并且会以监听模式自动运行 webpack，在浏览器打开<code>http://localhost:8080/</code>或 <code>http://localhost:8080/webpack-dev-server/</code> 可以浏览项目中的页面和<br>编译后的资源输出，并且通过一个 <code>socket.io</code> 服务实时监听它们的变化并自动刷新页面</p>
</blockquote>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line"># 安装</span><br><span class="line">$ npm install webpack-dev-server -g</span><br><span class="line"># 运行</span><br><span class="line">$ webpack-dev-server --progress --colors</span><br></pre></td></tr></table></figure>
<h2 id="七、故障处理"><a href="#七、故障处理" class="headerlink" title="七、故障处理"></a>七、故障处理</h2><hr>
<ul>
<li><code>Webpack</code> 的配置比较复杂，很容出现错误，下面是一些通常的故障处理手段</li>
<li>一般情况下，<code>webpack</code>如果出问题，会打印一些简单的错误信息，比如模块没有找到。我们还可以通过参数 <code>--display-error-details</code>来打印错误详情</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">$ webpack --display-error-details</span><br><span class="line">Hash: a40fbc6d852c51fceadb</span><br><span class="line">Version: webpack 1.12.2</span><br><span class="line">Time: 586ms</span><br><span class="line">Asset Size Chunks Chunk Names</span><br><span class="line">bundle.js 12.1 kB 0 [emitted] main</span><br><span class="line">[0] ./entry.js 153 bytes &#123;0&#125; [built] [1 error]</span><br><span class="line">[5] ./module.js 43 bytes &#123;0&#125; [built]</span><br><span class="line">+ 4 hidden modules</span><br><span class="line">ERROR in ./entry.js</span><br><span class="line">Module not found: Error: Cannot resolve &apos;file&apos; or &apos;directory&apos; ./badpathmodule in /Users/zhaoda/data/projects/webpack-handbook/examples</span><br><span class="line">resolve file</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule doesn&apos;t exist</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.webpack.js doesn&apos;t exist</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.js doesn&apos;t exist</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.web.js doesn&apos;t exist</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.json doesn&apos;t exist</span><br><span class="line">resolve directory</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule doesn&apos;t exist (directory default file)</span><br><span class="line">/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule/package.json doesn&apos;t exist (directory description file)</span><br><span class="line">[/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule]</span><br><span class="line">[/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.webpack.js]</span><br><span class="line">[/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.js]</span><br><span class="line">[/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.web.js]</span><br><span class="line">[/Users/zhaoda/data/projects/webpack-handbook/examples/badpathmodule.json]</span><br><span class="line">@ ./entry.js 3:0-26</span><br></pre></td></tr></table></figure>
<ul>
<li><code>Webpack</code>的配置提供了 <code>resolve</code> 和<code>resolveLoader</code> 参数来设置模块解析的处理细节， <code>resolve</code>用来配置应用层的模块（要被打包的模块）解析， <code>resolveLoader</code>用来配置<code>loader</code>模块的解析</li>
<li>当引入通过 <code>npm</code> 安装的 <code>node.js</code> 模块时，可能出现找不到依赖的错误。<code>Node.js</code>模块的依赖解析算法很简单，是通过查看模块的每一层父目录中的 <code>node_modules</code> 文件夹来查询依赖的。当出现 <code>Node.js</code>模块依赖查找失败的时候，可以尝试设置 <code>resolve.fallback</code>和<code>resolveLoader.fallback</code> 来解决问题</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">module.exports = &#123;</span><br><span class="line">resolve: &#123; fallback: path.join(__dirname, &quot;node_modules&quot;) &#125;,</span><br><span class="line">resolveLoader: &#123; fallback: path.join(__dirname, &quot;node_modules&quot;) &#125;</span><br><span class="line">&#125;;</span><br></pre></td></tr></table></figure>
<blockquote>
<p><code>Webpack</code>中涉及路径配置最好使用绝对路径，建议通过 <code>path.resolve(__dirname,&quot;app/folder&quot;)</code> 或 <code>path.join(__dirname, &quot;app&quot;, &quot;folder&quot;)</code>的方式来配置，以兼容 <code>Windows</code>环境</p>
</blockquote>
<h2 id="八、开发和部署技巧"><a href="#八、开发和部署技巧" class="headerlink" title="八、开发和部署技巧"></a>八、开发和部署技巧</h2><hr>
<h3 id="8-1-启用source-map"><a href="#8-1-启用source-map" class="headerlink" title="8.1 启用source-map"></a>8.1 启用source-map</h3><hr>
<ul>
<li>现在的代码是合并以后的代码，不利于排错和定位，只需要在config中添加</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">devtool: &apos;eval-source-map&apos;,</span><br></pre></td></tr></table></figure>
<ul>
<li>这样出错以后就会采用source-map的形式直接显示你出错代码的位置</li>
</ul>
<h3 id="8-2-使用preLoaders和postLoaders"><a href="#8-2-使用preLoaders和postLoaders" class="headerlink" title="8.2 使用preLoaders和postLoaders"></a>8.2 使用preLoaders和postLoaders</h3><hr>
<ul>
<li>也许你想在写代码的时候检查自己的<code>js</code>是否符合<code>jshint</code>的规范，那么隆重推荐<code>preLoaders</code>和<code>postLoaders</code></li>
<li><p><code>perLoaders</code>顾名思义就是在<code>loaders</code>执行之前处理的，<code>webpack</code>的处理顺序是<code>perLoaders - loaders - postLoaders</code></p>
</li>
<li><p>安装<code>jshint</code></p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm install jshint-loader --save-dev</span><br></pre></td></tr></table></figure>
<ul>
<li>在<code>config</code>文件中配置</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">module: &#123;</span><br><span class="line">...</span><br><span class="line">//和loaders一样的语法，很简单</span><br><span class="line">perLoaders: [</span><br><span class="line">&#123;</span><br><span class="line">test: /\.jsx?$/,</span><br><span class="line">include: APP_PATH,</span><br><span class="line">loader: &apos;jshint-loader&apos;</span><br><span class="line">&#125;</span><br><span class="line">]</span><br><span class="line">&#125;</span><br><span class="line">...</span><br><span class="line">//配置jshint的选项，支持es6的校验</span><br><span class="line">jshint: &#123;</span><br><span class="line">&quot;esnext&quot;: true</span><br><span class="line">&#125;,</span><br></pre></td></tr></table></figure>
<ul>
<li>好了 现在每次npm run start的时候就可以看到jshint的提示信息啦</li>
</ul>
<h3 id="8-3-部署上线"><a href="#8-3-部署上线" class="headerlink" title="8.3 部署上线"></a>8.3 部署上线</h3><hr>
<ul>
<li><p>刚才说的各种情况都是在开发时候的情况，那么假如项目已经开发完了，需要部署上线了。我们应该新创建一个单独的<code>config</code>文件，因为部署上线使用<code>webpack</code>的时候我们不需要一些<code>dev-tools</code>,<code>dev-server</code>和<code>jshint</code>校验等</p>
</li>
<li><p>复制我们现有的<code>config</code>文件，命名<code>webpack.production.config.js</code>，将里面关于 <code>devServer</code>等和开发有关的东西删掉</p>
</li>
<li><p>在<code>package.json</code>中添加一个命令</p>
</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">&quot;scripts&quot;: &#123;</span><br><span class="line">&quot;start&quot;: &quot;webpack-dev-server --hot --inline&quot;,</span><br><span class="line">&quot;build&quot;: &quot;webpack --progress --profile --colors --config webpack.production.config.js&quot;</span><br><span class="line">&#125;,</span><br></pre></td></tr></table></figure>
<ul>
<li>当要上线的时候,运行</li>
</ul>
<figure class="highlight plain"><table><tr><td class="code"><pre><span class="line">npm run build</span><br></pre></td></tr></table></figure>
<ul>
<li>可以发现<code>build</code>文件夹中生成了所有东西</li>
</ul>

      </div>
    
  </div>

</article>

<button class="assist-btn2 circle" id="assist_btn2" title="点亮屏幕" style="left: 27px; top: 152px;">
  <i class="iconfont" style="display:inline-block;color:red;width:20px;height:20px;">&#xe61d;</i>
</button>
<button class="assist-btn1 circle" id="assist_btn1" title="关闭屏幕亮度" style="left: 27px; top: 152px;">
  <i class="iconfont toc-title" style="display:inline-block;color:red;width:20px;height:20px;">&#xe61d;</i>
</button>


<script src="//cdn.bootcss.com/jquery/3.1.1/jquery.min.js"></script>	

<script src="https://my.openwrite.cn/js/readmore.js" type="text/javascript"></script>
<script>
  const btw = new BTWPlugin();
  btw.init({
    id: "container",
    blogId: "22699-1592137983091-414",
    name: "前端进阶之旅",
    qrcode: "https://poetries1.gitee.io/img-repo/2020/06/qrcode.jpg",
    keyword: "3a3b3c",
  });
</script>

<script type="text/javascript">

// white theme
var body = {color: "#555", background: "#000"};
var a_tag = {color: "#222"};
var header = { background: "#222"};
var logo_line_i = {background: "#222"};
// var post_code = {background: "#eee", color: "#222"};

function switch_theme() {
 $("body").css(body);
 $("a:not('.links-of-author-item a, .site-state-item a, .site-state-posts a, .feed-link a, .motion-element a, .post-tags a, .show-commit-cls a, #donate_board a')").css(a_tag);
 $(".header, .footer").css(header);
 $(".logo-line-before i, .logo-line-after i").css(logo_line_i);
 //$(".post code").css(post_code);
 $("#idhyt-surprise-ball #idhyt-surprise-ball-animation .drag").css(a_tag);
 $(".post-title-link, .posts-expand .post-meta, .post-comments-count, .disqus-comment-count, .post-category a, .post-nav-next a, .post-nav-item a").css(a_tag);
 
 // $("code").css({color: '#c5c8c6', background: '#1d1f21'});
 //$("#assist_btn1").hide(1500);
}

$(function () {
$("#assist_btn2").css("display","none");
 $("#assist_btn1").click(function() {
     switch_theme();
$("div#toc.toc-article").css({
 "background":"#eaeaea",
 "opacity":1
});
$(".toc-article ol").show();
$("#toc.toc-article .toc-title").css("color","#a98602");
$("#assist_btn1").css("display","none");
$("#assist_btn2").css("display","block");
 });
$("#assist_btn2").click(function() {
$("#assist_btn2").css("display","none");
$("#assist_btn1").css("display","block");
$("body").css("background","url(http://www.miaov.com/static/ie/images/news/bg.png)")
     $(".header, .footer").css("background","url(http://www.miaov.com/static/ie/images/news/bg.png)")
$(".toc-article ol").toggle(1000);
 });
});


//背景随机

var Y, O, E, L, B, C, T, z, N, S, A, I;
!function() {
var e = function() {
for (O.clearRect(0, 0, L, B), T = [{
x: 0,
y: .7 * B + C
}, {
x: 0,
y: .7 * B - C
}]; T[1].x < L + C;) t(T[0], T[1])
}, t = function(e, t) {
O.beginPath(), O.moveTo(e.x, e.y), O.lineTo(t.x, t.y);
var n = t.x + (2 * I() - .25) * C,
 r = a(t.y);
O.lineTo(n, r), O.closePath(), N -= S / -50, O.fillStyle = "#" + (127 * A(N) + 128 << 16 | 127 * A(N + S / 3) + 128 << 8 | 127 * A(N + S / 3 * 2) + 128).toString(16), O.fill(), T[0] = T[1], T[1] = {
 x: n,
 y: r
}
}, a = function n(e) {
var t = e + (2 * I() - 1.1) * C;
return t > B || t < 0 ? n(e) : t
};
Y = document.getElementById("evanyou"), O = Y.getContext("2d"), E = window.devicePixelRatio || 1, L = window.innerWidth, B = window.innerHeight, C = 90, z = Math, N = 0, S = 2 * z.PI, A = z.cos, I = z.random, Y.width = L * E, Y.height = B * E, O.scale(E, E), O.globalAlpha = .6, document.onclick = e, document.ontouchstart = e, e()
}()

   
$("#toc-eye").click(function(){
$("#toc.toc-article").toggle(1000);
});

</script>


   
  <div class="text-center donation">
    <div class="inner-donation">
      <span class="btn-donation">支持一下</span>
      <div class="donation-body">
        <div class="tip text-center">扫一扫，支持poetries</div>
        <ul>
        
          <li class="item">
            
              <span>微信扫一扫</span>
            
            <img src="/images/weixin.jpg" alt="">
          </li>
        
          <li class="item">
            
              <span>支付宝扫一扫</span>
            
            <img src="/images/zhifubao.jpg" alt="">
          </li>
        
        </ul>
      </div>
    </div>
  </div>


   
  <div class="box-prev-next clearfix">
    <a class="show pull-left" href="/2017/01/14/js-and-jquery-dom-compare/">
        <i class="icon icon-angle-left"></i>
    </a>
    <a class="show pull-right" href="/2017/01/21/gulp-works/">
        <i class="icon icon-angle-right"></i>
    </a>
  </div>




</div>


  <a id="backTop" class="back-top">
    <i class="icon-angle-up"></i>
  </a>




  <div class="modal" id="modal">
  <span id="cover" class="cover hide"></span>
  <div id="modal-dialog" class="modal-dialog hide-dialog">
    <div class="modal-header">
      <span id="close" class="btn-close">关闭</span>
    </div>
    <hr>
    <div class="modal-body">
      <ul class="list-toolbox">
        
          <li class="item-toolbox">
            <a
              class="CIRCLE"
              href="/archives/"
              rel="noopener noreferrer"
              target="_self"
              >
              博客
            </a>
          </li>
        
          <li class="item-toolbox">
            <a
              class="CIRCLE"
              href="/categories/"
              rel="noopener noreferrer"
              target="_self"
              >
              分类
            </a>
          </li>
        
          <li class="item-toolbox">
            <a
              class="CIRCLE"
              href="/tags/"
              rel="noopener noreferrer"
              target="_self"
              >
              标签
            </a>
          </li>
        
          <li class="item-toolbox">
            <a
              class="CIRCLE"
              href="/search/"
              rel="noopener noreferrer"
              target="_self"
              >
              搜索
            </a>
          </li>
        
          <li class="item-toolbox">
            <a
              class="CIRCLE"
              href="/link/"
              rel="noopener noreferrer"
              target="_self"
              >
              友链
            </a>
          </li>
        
          <li class="item-toolbox">
            <a
              class="CIRCLE"
              href="/about/"
              rel="noopener noreferrer"
              target="_self"
              >
              关于
            </a>
          </li>
        
      </ul>

    </div>
  </div>
</div>



  
      <div class="fexo-comments comments-post">
    

    

    
    

    

    
    

    

<!-- Gitalk评论插件通用代码 -->
<div id="gitalk-container"></div>

<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/gitalk@1/dist/gitalk.css">
<script src="https://cdn.jsdelivr.net/npm/gitalk@1/dist/gitalk.min.js"></script>
<script>
const gitalk = new Gitalk({
  clientID: '5567a2c4abb858009d96',
  clientSecret: 'b9039ec056cf5c2346b3cdb63308a28c163f91e5',
  repo: 'poetries.github.io',
  owner: 'poetries',
  // 在这里设置一下截取前50个字符串, 这是因为 github 对 label 的长度有了要求, 如果超过
  // 50个字符串则会报错.
  // id: location.pathname.split('/').pop().substring(0, 49),
  id: location.pathname,
  admin: ['poetries'],
  // facebook-like distraction free mode
  distractionFreeMode: false
})
gitalk.render('gitalk-container')
</script>
<!-- Gitalk代码结束 -->



  </div>

  

  <script type="text/javascript">
  function loadScript(url, callback) {
    var script = document.createElement('script')
    script.type = 'text/javascript';

    if (script.readyState) { //IE
      script.onreadystatechange = function() {
        if (script.readyState == 'loaded' ||
          script.readyState == 'complete') {
          script.onreadystatechange = null;
          callback();
        }
      };
    } else { //Others
      script.onload = function() {
        callback();
      };
    }

    script.src = url;
    document.getElementsByTagName('head')[0].appendChild(script);
  }

  window.onload = function() {
    loadScript('/js/bundle.js?235683', function() {
      // load success
    });
  }
</script>


  <!-- 页面点击小红心 -->
  <script type="text/javascript" src="/js/clicklove.js"></script>
 
  
</body>
</html>
